### Cartilla de Referencia - OrgaSmallI

# Procesador OrgaSmallI

OrgaSmallI es un procesador diseñado e implementado sobre la herramienta *Logisim*. Cuenta con las mismas características que el procesador **OrgaSmall** más un controlador de interrupciones, nuevas instrucciones y señales:

- Nuevo flag para indicar si las interrupciones están habilitadas (flag\_I), que comienza en cero.
- Dirección reservada: 0xFF, donde se almacenará la dirección de retorno antes de comenzar la rutina de la interrupción.

# ¿Qué pasa cuando interrumpen al CPU?

En el caso de un procesador OrgaSmallI, si el dispositivo de E/S activa la señal de interrupción y el flag I vale 1, termina de ejecutar la instrucción en curso y realiza atómicamente los siguientes pasos:

- Coloca [0xFF] = PC
- Coloca I=0 para evitar que el procesador vuelva a interrumpirse
- Coloca PC = [0x00]
- Activa la señal INTA para indicarle al dispositivo que atenderá su pedido

Luego, comienza a ejecutarse la rutina de atención de la interrupción propiamente dicha.

### Instrucciones

| Instrucción   | CodOp | Formato | Acción                                                      |  |  |  |  |
|---------------|-------|---------|-------------------------------------------------------------|--|--|--|--|
| ADD Rx, Ry    | 00001 | A       | $Rx \leftarrow Rx + Ry$                                     |  |  |  |  |
| ADC Rx, Ry    | 00010 | A       | Rx ← Rx + Ry + flag_C                                       |  |  |  |  |
| SUB Rx, Ry    | 00011 | A       | $Rx \leftarrow Rx - Ry$                                     |  |  |  |  |
| AND Rx, Ry    | 00100 | A       | $Rx \leftarrow Rx$ and $Ry$                                 |  |  |  |  |
| OR Rx, Ry     | 00101 | A       | $Rx \leftarrow Rx \text{ or } Ry$                           |  |  |  |  |
| XOR Rx, Ry    | 00110 | A       | Rx ← Rx xor Ry                                              |  |  |  |  |
| CMP Rx, Ry    | 00111 | A       | Modifica flags de Rx - Ry                                   |  |  |  |  |
| MOV Rx, Ry    | 01000 | A       | Rx ← Ry                                                     |  |  |  |  |
| SIG Rx        | 01001 | В       | $Rx \leftarrow Rx + 1$                                      |  |  |  |  |
| NEG Rx        | 01010 | В       | $Rx \leftarrow 0 - Rx$                                      |  |  |  |  |
| MIX Rx, Ry    | 01011 | A       | $  \mathbf{Rx} \leftarrow B_1 A_6 B_3 A_4 B_5 A_2 B_7 A_0 $ |  |  |  |  |
| STR [M], Rx   | 10000 | D       | Mem[M] ← Rx                                                 |  |  |  |  |
| LOAD Rx, [M]  | 10001 | D       | $\mathtt{Rx} \leftarrow \mathtt{Mem}[\mathtt{M}]$           |  |  |  |  |
| STR [Rx], Ry  | 10010 | A       | Mem[Rx] ← Ry                                                |  |  |  |  |
| LOAD Rx, [Ry] | 10011 | A       | $\mathtt{Rx} \leftarrow \mathtt{Mem}[\mathtt{Ry}]$          |  |  |  |  |
| JMP M         | 10100 | C       | $\mathtt{PC} \leftarrow \mathtt{M}$                         |  |  |  |  |
| JC M          | 10101 | C       | Si flag_C=1 entonces $PC \leftarrow M$                      |  |  |  |  |
| JZ M          | 10110 | C       | Si flag_Z=1 entonces $PC \leftarrow M$                      |  |  |  |  |
| JN M          | 10111 | C       | Si flag_N=1 entonces $PC \leftarrow M$                      |  |  |  |  |
| INC Rx        | 11000 | В       | $Rx \leftarrow Rx + 1$                                      |  |  |  |  |
| DEC Rx        | 11001 | В       | $Rx \leftarrow Rx - 1$                                      |  |  |  |  |
| SHR Rx, t     | 11010 | A       | $Rx \leftarrow Rx \ll t$                                    |  |  |  |  |
| SHL Rx, t     | 11011 | A       | $Rx \leftarrow Rx >> t$                                     |  |  |  |  |
| STI M         | 11100 | E       | flag_I ← 1                                                  |  |  |  |  |
| CLI M         | 11101 | E       | flag_I ← 0                                                  |  |  |  |  |
| IRET M        | 11110 | E       | $PC \leftarrow [0xFF], flag_I \leftarrow 1$                 |  |  |  |  |
| SET Rx, M     | 11111 | D       | $Rx \leftarrow M$                                           |  |  |  |  |

Las instrucciones están codificadas en 16 bits. Los primeros 5 bits son el opcode de la instrucción, el resto de los bits indican los parámetros. Existen 4 posibles codificaciones de parámetros:

| Formato      | Codificación       | XXX codifica el número del registro X (Rx), vale entre 0 y |
|--------------|--------------------|------------------------------------------------------------|
| A            | CCCCC XXX YYY      | 1.                                                         |
| $\mathbf{B}$ | CCCCC XXX          | YYY codifica el número del registro Y (Ry) o un inmediato, |
| $\mathbf{C}$ | CCCCC MMMMMMMM     | vale entre 0 y 7.                                          |
| D            | CCCCC XXX MMMMMMMM | A CA                   |
| $\mathbf{E}$ | CCCCC              | MMMMMMM Dirección de memoria o valor inmediato,            |
|              | I                  | número de 8 bits.                                          |

Las posiciones indicadas con – deben ser cero. Las instrucciones de opcode: 12, 13, 14, 15 son instrucciones reservadas.

#### Micro-instrucciones

Los datos almacenados en la memoria de micro-instrucciones corresponden a señales para los distintos componentes del sistema. La siguiente tabla indica a qué bit de micro-instrucción corresponde cada señal.



La codificación de las operaciones de la unidad artimético lógica es la siguiente:

| Código | Operación | Código | Operación | Código | Operación | Código | Operación |
|--------|-----------|--------|-----------|--------|-----------|--------|-----------|
| 0000   | Reservada | 0100   | AND       | 1000   | SHR       | 1100   | cte0x00   |
| 0001   | ADD       | 0101   | OR        | 1001   | SHL       | 1101   | cte0x01   |
| 0010   | ADC       | 0110   | XOR       | 1010   | _         | 1110   | cte0x02   |
| 0011   | SUB       | 0111   | CMP       | 1011   | MIX       | 1111   | cte0xFF   |

## Herramientas

### Ensamblador (assembler.py)

El ensamblador toma como entrada un archivo de texto con la lista de mnemónicos de instrucciones y genera el código binario del programa. Las instrucciones soportadas son todas las descriptas por la arquitectura.

Además el ensamblador soporta el uso de etiquetas y comentarios. Las etiquetas pueden ser cualquier cadena de caracteres finalizada en ":". Una vez declarada, puede ser utilizada en cualquier parte del código, incluso como parámetro o valor inmediato. Para declarar un comentario, se utiliza el caracter ";". Todo texto luego de la primera aparición de éste será considerado comentario.

El ensamblador también soporta la declaración de valores inmediatos. Para esto se utiliza la palabra reservada DB seguida de un valor numérico inmediato. Éste será incluido directamente en el código generado.

### Generador de Micro-instrucciones (buildMicroOps.py)

El generador de micro-instrucciones toma como entrada un archivo de descripción de señales que respeta la siguiente sintaxis:

```
<binary_opcode>:
<signal_1> <signal_2> ... <signal_n>
...
<signal_1> <signal_2> ... <signal_n>
```

donde <br/> donde <br/> signal> corresponde al valor de opcode a codificar en binario y <signal> a<br/> un nombre de señal. Las señales pueden ser indicadas por el nombre de la misma o como:<br/> <signal>=<x> donde x indica el valor de la señal.

Para señales de más de un bit, como ALU\_OP se debe utilizar el número entero decimal correspondiente. En particular, para la señal mencionada, se puede utilizar directamente el nombre de la operación en la ALU. Si no se indica el valor que debe tomar la señal, se utiliza 1. Por ejemplo, para la codificación de la instrucción ADD:

```
O0001: ; ADD

RB_enOut ALU_enA RB_selectIndexOut=0 ; ALU_A := Rx

RB_enOut ALU_enB RB_selectIndexOut=1 ; ALU_B := Ry

ALU_OP=ADD ALU_opW ; ALU_ADD

RB_enIn ALU_enOut RB_selectIndexIn=0 ; Rx := ALU_OUT

reset_microOp
```

Notar que la señal ALU\_OP es completada con un texto que indica la operación de la ALU a realizar.

#### Uso

Una vez dentro de *Logisim* se debe cargar el circuito de OrgaSmallI. Para cargar un programa, se debe entrar en el circuito memory y una vez seleccionada la memoria, utilizar el comando *load* para cargar un nuevo programa.

Para modificar el código de micro-instrucciones, se debe entrar al circuito controlUnit y con el mismo procedimiento anterior, cargar el nuevo código de micro-instrucciones. Recordar que el PC comienza siempre en cero y apunta a la primera instrucción a ejecutar.

Nota: se deberá guardar la dirección de la rutina de la interrupción en la dirección 0x00 antes de habilitar las interrupciones.